home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 16 / CU Amiga Magazine's Super CD-ROM 16 (1997-10-16)(EMAP Images)(GB)[!][issue 1997-11].iso / CUCD / Graphics / Ghostscript / source / gdevbit.c < prev    next >
C/C++ Source or Header  |  1996-09-09  |  11KB  |  342 lines

  1. /* Copyright (C) 1991, 1995, 1996 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* gdevbit.c */
  20. /* "Plain bits" devices to measure rendering time. */
  21. #include "gdevprn.h"
  22. #include "gsparam.h"
  23. #include "gxlum.h"
  24.  
  25. /*
  26.  * When debugging problems on large CMYK devices, we have to be able to
  27.  * modify the output at this stage.  These parameters are not set in any
  28.  * normal configuration.
  29.  */
  30. /* Define whether to trim off top and bottom white space. */
  31. /*#define TRIM_TOP_BOTTOM*/
  32. /* Define left and right trimming margins. */
  33. /* Note that this is only approximate: we trim to byte boundaries. */
  34. /*#define TRIM_LEFT 400*/
  35. /*#define TRIM_RIGHT 400*/
  36. /* Define whether to expand each bit to a byte. */
  37. /* Also convert black-and-white to proper gray (inverting if monobit). */
  38. /*#define EXPAND_BITS_TO_BYTES*/
  39.  
  40. /* Define the device parameters. */
  41. #ifndef X_DPI
  42. #  define X_DPI 72
  43. #endif
  44. #ifndef Y_DPI
  45. #  define Y_DPI 72
  46. #endif
  47.  
  48. /* The device descriptor */
  49. private dev_proc_map_rgb_color(bit_mono_map_rgb_color);
  50. private dev_proc_map_rgb_color(bit_map_rgb_color);
  51. private dev_proc_map_color_rgb(bit_map_color_rgb);
  52. private dev_proc_map_cmyk_color(bit_map_cmyk_color);
  53. private dev_proc_put_params(bit_put_params);
  54. private dev_proc_print_page(bit_print_page);
  55. #define bit_procs(map_rgb_color, map_cmyk_color)\
  56. {    gdev_prn_open,\
  57.     gx_default_get_initial_matrix,\
  58.     NULL,    /* sync_output */\
  59.     gdev_prn_output_page,\
  60.     gdev_prn_close,\
  61.     map_rgb_color,\
  62.     bit_map_color_rgb,\
  63.     NULL,    /* fill_rectangle */\
  64.     NULL,    /* tile_rectangle */\
  65.     NULL,    /* copy_mono */\
  66.     NULL,    /* copy_color */\
  67.     NULL,    /* draw_line */\
  68.     NULL,    /* get_bits */\
  69.     gdev_prn_get_params,\
  70.     bit_put_params,\
  71.     map_cmyk_color,\
  72.     NULL,    /* get_xfont_procs */\
  73.     NULL,    /* get_xfont_device */\
  74.     NULL,    /* map_rgb_alpha_color */\
  75.     gx_page_device_get_page_device    /* get_page_device */\
  76. }
  77.  
  78. private gx_device_procs bitmono_procs =
  79.   bit_procs(bit_mono_map_rgb_color, NULL);
  80. gx_device_printer far_data gs_bit_device =
  81. { prn_device_body(gx_device_printer, bitmono_procs, "bit",
  82.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  83.     X_DPI, Y_DPI,
  84.     0,0,0,0,            /* margins */
  85.     1,1,1,0,2,1, bit_print_page)
  86. };
  87.  
  88. private gx_device_procs bitrgb_procs =
  89.   bit_procs(bit_map_rgb_color, NULL);
  90. gx_device_printer far_data gs_bitrgb_device =
  91. { prn_device_body(gx_device_printer, bitrgb_procs, "bitrgb",
  92.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  93.     X_DPI, Y_DPI,
  94.     0,0,0,0,            /* margins */
  95.     3,4,1,1,2,2, bit_print_page)
  96. };
  97.  
  98. private gx_device_procs bitcmyk_procs =
  99.   bit_procs(NULL, bit_map_cmyk_color);
  100. gx_device_printer far_data gs_bitcmyk_device =
  101. { prn_device_body(gx_device_printer, bitcmyk_procs, "bitcmyk",
  102.     DEFAULT_WIDTH_10THS, DEFAULT_HEIGHT_10THS,
  103.     X_DPI, Y_DPI,
  104.     0,0,0,0,            /* margins */
  105.     4,4,1,1,2,2, bit_print_page)
  106. };
  107.  
  108. /* Map gray to color. */
  109. /* Note that 1-bit monochrome is a special case. */
  110. private gx_color_index
  111. bit_mono_map_rgb_color(gx_device *dev, gx_color_value red,
  112.   gx_color_value green, gx_color_value blue)
  113. {    int bpc = dev->color_info.depth;
  114.     int drop = sizeof(gx_color_value) * 8 - bpc;
  115.     gx_color_value gray = 
  116.       (red * (unsigned long)lum_red_weight +
  117.        green * (unsigned long)lum_green_weight +
  118.        blue * (unsigned long)lum_blue_weight +
  119.        (lum_all_weights / 2))
  120.         / lum_all_weights;
  121.     return (bpc == 1 ? gx_max_color_value - gray : gray) >> drop;
  122. }
  123.  
  124. /* Map RGB to color. */
  125. private gx_color_index
  126. bit_map_rgb_color(gx_device *dev, gx_color_value red,
  127.   gx_color_value green, gx_color_value blue)
  128. {    int bpc = dev->color_info.depth / 3;
  129.     int drop = sizeof(gx_color_value) * 8 - bpc;
  130.     return ((((red >> drop) << bpc) + (green >> drop)) << bpc) +
  131.       (blue >> drop);
  132. }
  133.  
  134. /* Map color to RGB.  This has 3 separate cases, but since it is rarely */
  135. /* used, we do a case test rather than providing 3 separate routines. */
  136. private int
  137. bit_map_color_rgb(gx_device *dev, gx_color_index color, gx_color_value rgb[3])
  138. {    int depth = dev->color_info.depth;
  139.     int ncomp = dev->color_info.num_components;
  140.     int bpc = depth / ncomp;
  141.     uint mask = (1 << bpc) - 1;
  142. #define cvalue(c) ((gx_color_value)((ulong)(c) * gx_max_color_value / mask))
  143.  
  144.     switch ( ncomp )
  145.     {
  146.     case 1:                /* gray */
  147.         rgb[0] = rgb[1] = rgb[2] =
  148.           (depth == 1 ? (color ? 0 : gx_max_color_value) :
  149.            cvalue(color));
  150.         break;
  151.     case 3:                /* RGB */
  152.         { gx_color_index cshift = color;
  153.           rgb[2] = cvalue(cshift & mask);  cshift >>= bpc;
  154.           rgb[1] = cvalue(cshift & mask);
  155.           rgb[0] = cvalue(cshift >> bpc);
  156.         }
  157.         break;
  158.     case 4:                /* CMYK */
  159.         /* Map CMYK back to RGB. */
  160.         { gx_color_index cshift = color;
  161.           uint c, m, y, k;
  162.           
  163.           k = cshift & mask;  cshift >>= bpc;
  164.           y = cshift & mask;  cshift >>= bpc;
  165.           m = cshift & mask;
  166.           c = cshift >> bpc;
  167.           /* We use our improved conversion rule.... */
  168.           rgb[0] = cvalue((mask - c) * (mask - k) / mask);
  169.           rgb[1] = cvalue((mask - m) * (mask - k) / mask);
  170.           rgb[2] = cvalue((mask - y) * (mask - k) / mask);
  171.         }
  172.         break;
  173.     }
  174.     return 0;
  175. #undef cvalue
  176. }
  177.  
  178. /* Map CMYK to color. */
  179. private gx_color_index
  180. bit_map_cmyk_color(gx_device *dev, gx_color_value cyan,
  181.   gx_color_value magenta, gx_color_value yellow, gx_color_value black)
  182. {    int bpc = dev->color_info.depth / 4;
  183.     int drop = sizeof(gx_color_value) * 8 - bpc;
  184.     gx_color_index color =
  185.       ((((((cyan >> drop) << bpc) +
  186.           (magenta >> drop)) << bpc) +
  187.         (yellow >> drop)) << bpc) +
  188.       (black >> drop);
  189.  
  190.     return (color == gx_no_color_index ? color ^ 1 : color);
  191. }
  192.  
  193. /* Set parameters.  We allow setting the number of bits per component. */
  194. private int
  195. bit_put_params(gx_device *pdev, gs_param_list *plist)
  196. {    gx_device_color_info save_info;
  197.     int ncomps = pdev->color_info.num_components;
  198.     int bpc = pdev->color_info.depth / ncomps;
  199.     int v;
  200.     int ecode = 0;
  201.     int code;
  202.     static const byte depths[4][8] = {
  203.       { 1, 2, 0, 4, 8, 0, 0, 8 },
  204.       { 0 },
  205.       { 4, 8, 0, 16, 16, 0, 0, 24 },
  206.       { 4, 8, 0, 16, 32, 0, 0, 32 }
  207.     };
  208.     const char _ds *vname;
  209.  
  210.     if ( (code = param_read_int(plist, (vname = "GrayValues"), &v)) != 1 ||
  211.          (code = param_read_int(plist, (vname = "RedValues"), &v)) != 1 ||
  212.          (code = param_read_int(plist, (vname = "GreenValues"), &v)) != 1 ||
  213.          (code = param_read_int(plist, (vname = "BlueValues"), &v)) != 1
  214.        )
  215.     {    if ( code < 0 )
  216.           ecode = code;
  217.         else switch ( v )
  218.         {
  219.         case 2: bpc = 1; break;
  220.         case 4: bpc = 2; break;
  221.         case 16: bpc = 4; break;
  222.         case 32: bpc = 5; break;
  223.         case 256: bpc = 8; break;
  224.         default: param_signal_error(plist, vname,
  225.                         ecode = gs_error_rangecheck);
  226.         }
  227.     }
  228.  
  229.     if ( ecode < 0 )
  230.       return ecode;
  231.     /* Temporarily reset the color_info so that gdev_prn_put_params */
  232.     /* won't complain. */
  233.     save_info = pdev->color_info;
  234.     if ( code != 1 )
  235.       {    pdev->color_info.depth = depths[ncomps - 1][bpc - 1];
  236.         pdev->color_info.max_gray = pdev->color_info.max_color =
  237.           (pdev->color_info.dither_grays =
  238.            pdev->color_info.dither_colors =
  239.              (1 << bpc)) - 1;
  240.       }
  241.     ecode = gdev_prn_put_params(pdev, plist);
  242.     if ( ecode < 0 )
  243.       {    pdev->color_info = save_info;
  244.         return ecode;
  245.       }
  246.     if ( code != 1 )
  247.       {    if ( pdev->is_open )
  248.           gs_closedevice(pdev);
  249.       }
  250.     return 0;
  251. }
  252.  
  253. /* Send the page to the printer. */
  254. private int
  255. bit_print_page(gx_device_printer *pdev, FILE *prn_stream)
  256. {    /* Just dump the bits on the file. */
  257.     /* If the file is 'nul', don't even do the writes. */
  258.     int line_size = gdev_mem_bytes_per_scan_line((gx_device *)pdev);
  259.     byte *in = (byte *)gs_malloc(line_size, 1, "bit_print_page(in)");
  260.     byte *data;
  261.     int nul = !strcmp(pdev->fname, "nul");
  262.     int lnum = 0, bottom = pdev->height;
  263.  
  264.     if ( in == 0 )
  265.       return_error(gs_error_VMerror);
  266. #ifdef TRIM_TOP_BOTTOM
  267.     { gx_color_index white =
  268.         (pdev->color_info.num_components == 4 ? 0 :
  269.          (*dev_proc(pdev, map_rgb_color))
  270.            ((gx_device *)pdev, gx_max_color_value, gx_max_color_value,
  271.         gx_max_color_value));
  272. #define color_index_bits (sizeof(gx_color_index) * 8)
  273.       const gx_color_index *p;
  274.       const gx_color_index *end;
  275.       int depth = pdev->color_info.depth;
  276.       int end_bits = ((long)line_size * depth) % color_index_bits;
  277.       gx_color_index end_mask =
  278.         (end_bits == 0 ? 0 : -1 << (color_index_bits - end_bits));
  279.       int i;
  280.  
  281.       for ( i = depth; i < color_index_bits; i += depth )
  282.         white |= white << depth;
  283.       /* Remove bottom white space. */
  284.       for ( ; bottom - lnum > 1; --bottom )
  285.         { gdev_prn_get_bits(pdev, bottom - 1, in, &data);
  286.           p = (const gx_color_index *)data;
  287.           end = p + line_size / sizeof(gx_color_index);
  288.           for ( ; p < end; ++p )
  289.         if ( *p != white )
  290.           goto bx;
  291.           if ( end_mask != 0 && ((*end ^ white) & end_mask) != 0 )
  292.         goto bx;
  293.         }
  294.       /* Remove top white space. */
  295. bx:      for ( ; lnum < bottom; ++lnum )
  296.         { gdev_prn_get_bits(pdev, lnum, in, &data);
  297.           p = (const gx_color_index *)data;
  298.           end = p + line_size / sizeof(gx_color_index);
  299.           for ( ; p < end; ++p )
  300.         if ( *p != white )
  301.           goto tx;
  302.           if ( end_mask != 0 && ((*end ^ white) & end_mask) != 0 )
  303.         goto tx;
  304.         }
  305. tx:      ;
  306.     }
  307. #endif
  308.     for ( ; lnum < bottom; ++lnum )
  309.       {    gdev_prn_get_bits(pdev, lnum, in, &data);
  310.         if ( !nul )
  311.           { const byte *row = data;
  312.             uint len = line_size;
  313. #ifdef TRIM_LEFT
  314.             row += (TRIM_LEFT * pdev->color_info.depth) >> 3;
  315. #endif
  316. #ifdef TRIM_RIGHT
  317.             len = data + ((TRIM_RIGHT * pdev->color_info.depth + 7) >> 3) - row;
  318. #endif
  319. #ifdef EXPAND_BITS_TO_BYTES
  320.             { uint i;
  321.               byte invert = (pdev->color_info.depth == 1 ? 0xff : 0);
  322.               for ( i = 0; i < len; ++i )
  323.             { byte b = row[i] ^ invert;
  324.               putc(-(b >> 7) & 0xff, prn_stream);
  325.               putc(-((b >> 6) & 1) & 0xff, prn_stream);
  326.               putc(-((b >> 5) & 1) & 0xff, prn_stream);
  327.               putc(-((b >> 4) & 1) & 0xff, prn_stream);
  328.               putc(-((b >> 3) & 1) & 0xff, prn_stream);
  329.               putc(-((b >> 2) & 1) & 0xff, prn_stream);
  330.               putc(-((b >> 1) & 1) & 0xff, prn_stream);
  331.               putc(-(b & 1) & 0xff, prn_stream);
  332.             }
  333.             }
  334. #else
  335.             fwrite(row, 1, len, prn_stream);
  336. #endif
  337.           }
  338.       }
  339.     gs_free((char *)in, line_size, 1, "bit_print_page(in)");
  340.     return 0;
  341. }
  342.